<!DOCTYPE html>
<html>
<head>
    

    

    



    <meta charset="utf-8">
    
    
    
    
    <title>设计模式（1）——单例模式详解 | 博客主页 | 世界是个球，前方总有路！</title>
    <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">
    
    <meta name="theme-color" content="#3F51B5">
    
    
    <meta name="keywords" content="设计模式">
    <meta name="description" content="一、前言&amp;emsp;&amp;emsp;之前过年在家，买了本《Head First设计模式》来看，看完后一直想写设计模式的系列博客，但是一直没开始。刚好今天看到《Java并发编程的艺术》这本书上对单例模式的两种多线程形式下的实现方式做了详细的介绍，让我对它们的实现机制有了更深入的了解，所以，借这个机会，来谈一谈单例模式。   二、正文&amp;emsp; 2.1 什么是单例模式&amp;emsp;&amp;emsp;单例模式是设">
<meta property="og:type" content="article">
<meta property="og:title" content="设计模式（1）——单例模式详解">
<meta property="og:url" content="http://tewuyiang.gitee.io/blog/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%EF%BC%881%EF%BC%89%E2%80%94%E2%80%94%E5%8D%95%E4%BE%8B%E6%A8%A1%E5%BC%8F%E8%AF%A6%E8%A7%A3/index.html">
<meta property="og:site_name" content="博客主页">
<meta property="og:description" content="一、前言&amp;emsp;&amp;emsp;之前过年在家，买了本《Head First设计模式》来看，看完后一直想写设计模式的系列博客，但是一直没开始。刚好今天看到《Java并发编程的艺术》这本书上对单例模式的两种多线程形式下的实现方式做了详细的介绍，让我对它们的实现机制有了更深入的了解，所以，借这个机会，来谈一谈单例模式。   二、正文&amp;emsp; 2.1 什么是单例模式&amp;emsp;&amp;emsp;单例模式是设">
<meta property="og:locale" content="zh_CN">
<meta property="article:published_time" content="2020-03-22T13:18:12.000Z">
<meta property="article:modified_time" content="2020-03-22T17:00:59.886Z">
<meta property="article:author" content="特务依昂">
<meta property="article:tag" content="设计模式">
<meta name="twitter:card" content="summary">
    
        <link rel="alternate" type="application/atom+xml" title="博客主页" href="/blog/atom.xml">
    
    <link rel="shortcut icon" href="/blog/img/title.png">
    <link rel="stylesheet" href="//unpkg.com/hexo-theme-material-indigo@latest/css/style.css">
    <script>window.lazyScripts=[]</script>

    <!-- custom head -->
    

<meta name="generator" content="Hexo 4.2.0"></head>

<body>
    <div id="loading" class="active"></div>

    <aside id="menu" class="hide" >
  <div class="inner flex-row-vertical">
    <a href="javascript:;" class="header-icon waves-effect waves-circle waves-light" id="menu-off">
        <i class="icon icon-lg icon-close"></i>
    </a>
    <div class="brand-wrap" style="background-image:url(/blog/img/brand.jpg)">
      <div class="brand">
        <a href="/blog/" class="avatar waves-effect waves-circle waves-light">
          <img src="/blog/img/avatar.jpg">
        </a>
        <hgroup class="introduce">
          <h5 class="nickname">特务依昂</h5>
          <a href="mailto:1131564805@qq.com" title="1131564805@qq.com" class="mail">1131564805@qq.com</a>
        </hgroup>
      </div>
    </div>
    <div class="scroll-wrap flex-col">
      <ul class="nav">
        
            <li class="waves-block waves-effect">
              <a href="/blog/"  >
                <i class="icon icon-lg icon-home"></i>
                主页
              </a>
            </li>
        
            <li class="waves-block waves-effect">
              <a href="/blog/archives"  >
                <i class="icon icon-lg icon-archives"></i>
                博客
              </a>
            </li>
        
            <li class="waves-block waves-effect">
              <a href="/blog/tags"  >
                <i class="icon icon-lg icon-tags"></i>
                标签
              </a>
            </li>
        
            <li class="waves-block waves-effect">
              <a href="/blog/categories"  >
                <i class="icon icon-lg icon-th-list"></i>
                分类
              </a>
            </li>
        
            <li class="waves-block waves-effect">
              <a href="https://github.com/tewuyiang" target="_blank" >
                <i class="icon icon-lg icon-github"></i>
                Github
              </a>
            </li>
        
            <li class="waves-block waves-effect">
              <a href="https://weibo.com/u/5516635708/" target="_blank" >
                <i class="icon icon-lg icon-weibo"></i>
                Weibo
              </a>
            </li>
        
            <li class="waves-block waves-effect">
              <a href="https://www.cnblogs.com/tuyang1129/" target="_blank" >
                <i class="icon icon-lg icon-link"></i>
                博客园
              </a>
            </li>
        
      </ul>
    </div>
  </div>
</aside>

    <main id="main">
        <header class="top-header" id="header">
    <div class="flex-row">
        <a href="javascript:;" class="header-icon waves-effect waves-circle waves-light on" id="menu-toggle">
          <i class="icon icon-lg icon-navicon"></i>
        </a>
        <div class="flex-col header-title ellipsis">设计模式（1）——单例模式详解</div>
        
        <div class="search-wrap" id="search-wrap">
            <a href="javascript:;" class="header-icon waves-effect waves-circle waves-light" id="back">
                <i class="icon icon-lg icon-chevron-left"></i>
            </a>
            <input type="text" id="key" class="search-input" autocomplete="off" placeholder="输入感兴趣的关键字">
            <a href="javascript:;" class="header-icon waves-effect waves-circle waves-light" id="search">
                <i class="icon icon-lg icon-search"></i>
            </a>
        </div>
        
        
        <a href="javascript:;" class="header-icon waves-effect waves-circle waves-light" id="menuShare">
            <i class="icon icon-lg icon-share-alt"></i>
        </a>
        
    </div>
</header>
<header class="content-header post-header">

    <div class="container fade-scale">
        <h1 class="title">设计模式（1）——单例模式详解</h1>
        <h5 class="subtitle">
            
                <time datetime="2020-03-22T13:18:12.000Z" itemprop="datePublished" class="page-time">
  2020-03-22
</time>


	<ul class="article-category-list"><li class="article-category-list-item"><a class="article-category-list-link" href="/blog/categories/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F/">设计模式</a></li></ul>

            
        </h5>
    </div>

    


</header>


<div class="container body-wrap">
    
    <aside class="post-widget">
        <nav class="post-toc-wrap post-toc-shrink" id="post-toc">
            <h4>TOC</h4>
            <ol class="post-toc"><li class="post-toc-item post-toc-level-2"><a class="post-toc-link" href="#一、前言"><span class="post-toc-number">1.</span> <span class="post-toc-text">一、前言</span></a></li><li class="post-toc-item post-toc-level-2"><a class="post-toc-link" href="#二、正文"><span class="post-toc-number">2.</span> <span class="post-toc-text">二、正文</span></a><ol class="post-toc-child"><li class="post-toc-item post-toc-level-3"><a class="post-toc-link" href="#emsp-2-1-什么是单例模式"><span class="post-toc-number">2.1.</span> <span class="post-toc-text">&amp;emsp; 2.1 什么是单例模式</span></a></li><li class="post-toc-item post-toc-level-3"><a class="post-toc-link" href="#emsp-2-2-饿汉式"><span class="post-toc-number">2.2.</span> <span class="post-toc-text">&amp;emsp; 2.2 饿汉式</span></a></li><li class="post-toc-item post-toc-level-3"><a class="post-toc-link" href="#emsp-2-3-懒汉式"><span class="post-toc-number">2.3.</span> <span class="post-toc-text">&amp;emsp; 2.3 懒汉式</span></a></li><li class="post-toc-item post-toc-level-3"><a class="post-toc-link" href="#emsp-2-4-双重校验锁"><span class="post-toc-number">2.4.</span> <span class="post-toc-text">&amp;emsp; 2.4 双重校验锁</span></a></li><li class="post-toc-item post-toc-level-3"><a class="post-toc-link" href="#emsp-2-5-静态内部类实现"><span class="post-toc-number">2.5.</span> <span class="post-toc-text">&amp;emsp; 2.5 静态内部类实现</span></a></li><li class="post-toc-item post-toc-level-3"><a class="post-toc-link" href="#emsp-2-6-枚举类实现"><span class="post-toc-number">2.6.</span> <span class="post-toc-text">&amp;emsp; 2.6 枚举类实现</span></a></li><li class="post-toc-item post-toc-level-3"><a class="post-toc-link" href="#emsp-2-7-单例模式的破坏"><span class="post-toc-number">2.7.</span> <span class="post-toc-text">&amp;emsp; 2.7 单例模式的破坏</span></a></li></ol></li><li class="post-toc-item post-toc-level-2"><a class="post-toc-link" href="#三、总结"><span class="post-toc-number">3.</span> <span class="post-toc-text">三、总结</span></a></li><li class="post-toc-item post-toc-level-2"><a class="post-toc-link" href="#四、参考"><span class="post-toc-number">4.</span> <span class="post-toc-text">四、参考</span></a></li></ol>
        </nav>
    </aside>


<article id="post-设计模式（1）——单例模式详解"
  class="post-article article-type-post fade" itemprop="blogPost">

    <div class="post-card">
        <h1 class="post-card-title">设计模式（1）——单例模式详解</h1>
        <div class="post-meta">
            <time class="post-time" title="2020-03-22 21:18:12" datetime="2020-03-22T13:18:12.000Z"  itemprop="datePublished">2020-03-22</time>

            
	<ul class="article-category-list"><li class="article-category-list-item"><a class="article-category-list-link" href="/blog/categories/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F/">设计模式</a></li></ul>



            
<span id="busuanzi_container_page_pv" title="文章总阅读量" style='display:none'>
    <i class="icon icon-eye icon-pr"></i><span id="busuanzi_value_page_pv"></span>
</span>


        </div>
        <div class="post-content" id="post-content" itemprop="postContent">
            <h2 id="一、前言"><a href="#一、前言" class="headerlink" title="一、前言"></a>一、前言</h2><p>&emsp;&emsp;之前过年在家，买了本<code>《Head First设计模式》</code>来看，看完后一直想写设计模式的系列博客，但是一直没开始。刚好今天看到<code>《Java并发编程的艺术》</code>这本书上对单例模式的两种多线程形式下的实现方式做了详细的介绍，让我对它们的实现机制有了更深入的了解，所以，借这个机会，来谈一谈单例模式。</p>
<br>

<h2 id="二、正文"><a href="#二、正文" class="headerlink" title="二、正文"></a>二、正文</h2><h3 id="emsp-2-1-什么是单例模式"><a href="#emsp-2-1-什么是单例模式" class="headerlink" title="&emsp; 2.1 什么是单例模式"></a>&emsp; 2.1 什么是单例模式</h3><p>&emsp;&emsp;单例模式是设计模式中比较简单，但是却使用非常广泛的一种。单例模式的定义如下：</p>
<blockquote>
<p><strong>确保一个类只有一个对象，并且提供这个对象的全局访问接口</strong>。</p>
</blockquote>
<p>&emsp;&emsp;这个应该很好理解，使用单例模式定义一个类，只能够通过它创建一个对象，并且提供一个接口，让全局都能获得这个对象。需要注意的是，这个类只有一个对象并不是一个约定，而是我们通过一些手段，强制让这个类只能有一个对象。而单例模式有五种经典的实现方式：</p>
<ol>
<li>饿汉式</li>
<li>懒汉式</li>
<li>双重校验锁</li>
<li>静态内部类实现</li>
<li>枚举类实现</li>
</ol>
<p>&emsp;&emsp;下面我就来一一介绍这五种实现方式。</p>
<br>

<h3 id="emsp-2-2-饿汉式"><a href="#emsp-2-2-饿汉式" class="headerlink" title="&emsp; 2.2 饿汉式"></a>&emsp; 2.2 饿汉式</h3><p>&emsp;&emsp;饿汉式应该是单例模式中最简单的一种实现方式了，我们直接看代码：</p>
<figure class="highlight java"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">Singleton</span> </span>&#123;</span><br><span class="line">    <span class="comment">// 单例对象</span></span><br><span class="line">    <span class="keyword">public</span> <span class="keyword">static</span> Singleton singleton = <span class="keyword">new</span> Singleton();</span><br><span class="line"></span><br><span class="line">    <span class="comment">// 构造器被私有</span></span><br><span class="line">    <span class="function"><span class="keyword">private</span> <span class="title">Singleton</span><span class="params">()</span> </span>&#123;</span><br><span class="line">    &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>

<p>&emsp;&emsp;上面的单例实现应该很好理解，在类中，我们直接初始化了一个<code>static</code>的对象，然后将构造器设为<code>private</code>，这样就没法在类的外部创建这个类的对象了，于是这个类就有了唯一的对象<code>singleton</code>。这也是饿汉式这个名字的由来——不等待，立即创建，像饿疯了一样（尬解）。而这个单例对象是<code>public</code>类型的，这也就保证了它能够被全局访问。</p>
<p>&emsp;&emsp;这种实现方式非常简单，而且在多线程的环境下也是安全的，因为类的静态成员是在类加载的时候被初始化，而类加载这个过程已经由<code>JVM</code>实现了线程同步，也就是说一个类不会被加载两次。但是这种实现方式也有缺陷，即单例对象是在类被加载的时候创建，假设我们需要创建的是一个非常复杂的单例对象，那这将占用大量资源，大大拖慢了类加载的效率。为了解决这个问题，于是出现了懒汉式。<br><br></p>
<h3 id="emsp-2-3-懒汉式"><a href="#emsp-2-3-懒汉式" class="headerlink" title="&emsp; 2.3 懒汉式"></a>&emsp; 2.3 懒汉式</h3><p>&emsp;&emsp;懒汉式的实现方式是，只有当单例对象第一次被用到的时候，才进行创建。这也是懒汉式名字的由来——懒得只有催它的时候才会动（再次尬解）。下面就来看看懒汉式的代码实现。</p>
<p>&emsp;<strong>（1）线程不安全实现</strong></p>
<p>&emsp;&emsp;在单线程中，懒汉式的实现如下：</p>
<figure class="highlight java"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">Singleton</span> </span>&#123;</span><br><span class="line">    <span class="comment">// 单例对象为private</span></span><br><span class="line">    <span class="keyword">private</span> <span class="keyword">static</span> Singleton singleton;</span><br><span class="line"></span><br><span class="line">    <span class="comment">// 构造器被私有</span></span><br><span class="line">    <span class="function"><span class="keyword">private</span> <span class="title">Singleton</span><span class="params">()</span> </span>&#123;</span><br><span class="line">    &#125;</span><br><span class="line"></span><br><span class="line">    <span class="comment">// 获取单例对象的方法</span></span><br><span class="line">    <span class="function"><span class="keyword">public</span> <span class="keyword">static</span> Singleton <span class="title">getSingleton</span><span class="params">()</span> </span>&#123;</span><br><span class="line">        <span class="comment">// 若单例对象还没创建，则先创建它</span></span><br><span class="line">        <span class="keyword">if</span> (<span class="keyword">null</span> == singleton)</span><br><span class="line">            singleton = <span class="keyword">new</span> Singleton();</span><br><span class="line">        <span class="keyword">return</span> singleton;</span><br><span class="line">    &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>

<p>&emsp;&emsp;上面的代码应该也很好理解，这次我们将单例对象的引用设置为私有，但是提供一个<code>public</code>的<code>get</code>方法来获取这个单例对象，也就是全局访问接口。在这个<code>get</code>方法中，我们先判断这个对象是否被创建，若没有被创建，则先创建它，这也就意味着这个对象将在第一次<code>get</code>时被创建，之后都会直接获取。同时，因为这个类的构造器是<code>private</code>的，所以这个类正常情况下无法创建其他对象。</p>
<p>&emsp;<strong>（2）线程安全实现</strong></p>
<p>&emsp;&emsp;上面这种实现方式，很好的解决了饿汉式拖慢加载速度的问题。但是它也有缺陷，那就是无法在多线程的情况下使用。在<code>getSingleton</code>方法中有三行代码，我们考虑在多线程中的情况会如何：假如有两条线程同时运行到了第一句代码，判断对象是否为空，此时对象还没有被创建，于是两条线程都会运行第二句代码，企图创建对象。第一条线程先创建了对象，赋值给了<code>singleton</code>，接着第二条线程也创建了一个新的对象，再次赋值给<code>singleton</code>，就将前面的引用给覆盖了。创建了多个对象，这就违反了单例模式的原则，为了解决这种情况，我们需要给这个方法进行多线程同步，也就是如下代码：</p>
<figure class="highlight java"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">Singleton</span> </span>&#123;</span><br><span class="line">    <span class="keyword">private</span> <span class="keyword">static</span> Singleton singleton;</span><br><span class="line">    <span class="function"><span class="keyword">private</span> <span class="title">Singleton</span><span class="params">()</span> </span>&#123;</span><br><span class="line">    &#125;</span><br><span class="line"></span><br><span class="line">    <span class="comment">// 使用synchronized关键字同步方法</span></span><br><span class="line">    <span class="function"><span class="keyword">public</span> <span class="keyword">synchronized</span> <span class="keyword">static</span> Singleton <span class="title">getSingleton</span><span class="params">()</span> </span>&#123;</span><br><span class="line">        <span class="keyword">if</span> (<span class="keyword">null</span> == singleton)</span><br><span class="line">            singleton = <span class="keyword">new</span> Singleton();</span><br><span class="line">        <span class="keyword">return</span> singleton;</span><br><span class="line">    &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>

<p>&emsp;&emsp;我们使用了<code>synchronized</code>关键字对方法进行了同步，这样一来，每次只会有一个线程执行方法，不会再出现上述情况了。但是，新的问题出现了，每条线程在调用<code>getSingleton</code>方法前，都需要先获得锁，这将大大影响运行的效率。而且我们发现，只有第一次创建对象时是需要同步的，当对象创建完成之后，这种同步就没有必要了，也就是说，对象创建完成后，这种同步完全就是没有意义的降低效率。为了解决这个问题，就有人想出了下面这种方式。</p>
<br>

<h3 id="emsp-2-4-双重校验锁"><a href="#emsp-2-4-双重校验锁" class="headerlink" title="&emsp; 2.4 双重校验锁"></a>&emsp; 2.4 双重校验锁</h3><p>&emsp;&emsp;还是一样，先来看双重校验锁的代码:</p>
<figure class="highlight java"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">Singleton</span> </span>&#123;</span><br><span class="line">    <span class="comment">// 需要加上volatile修饰</span></span><br><span class="line">    <span class="keyword">public</span> <span class="keyword">volatile</span> <span class="keyword">static</span> Singleton singleton;</span><br><span class="line">    </span><br><span class="line">    <span class="function"><span class="keyword">private</span> <span class="title">Singleton</span><span class="params">()</span> </span>&#123;</span><br><span class="line">    &#125;</span><br><span class="line"></span><br><span class="line">    <span class="function"><span class="keyword">public</span> <span class="keyword">static</span> Singleton <span class="title">getSingleton</span><span class="params">()</span> </span>&#123;</span><br><span class="line">        <span class="comment">// 1、先判断singleton是否被创建</span></span><br><span class="line">        <span class="keyword">if</span> (<span class="keyword">null</span> == singleton) &#123;</span><br><span class="line">            <span class="comment">// 2、开始创建对象，但是需要先同步</span></span><br><span class="line">            <span class="keyword">synchronized</span> (Singleton<span class="class">.<span class="keyword">class</span>) </span>&#123;</span><br><span class="line">                <span class="comment">// 3、在正式创建前，再次判断</span></span><br><span class="line">                <span class="keyword">if</span> (<span class="keyword">null</span> == singleton)</span><br><span class="line">                    singleton = <span class="keyword">new</span> Singleton();    <span class="comment">// 4、创建对象</span></span><br><span class="line">            &#125;</span><br><span class="line">        &#125;</span><br><span class="line">        <span class="keyword">return</span> singleton;</span><br><span class="line">    &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>

<p>&emsp;&emsp;双重校验锁的的代码和懒汉式类似，只是<code>getSingleton</code>的实现更加复杂，我们来看看它做了什么事情：</p>
<ol>
<li>第一步和懒汉式一样，先判断单例对象是否已经被创建，若被创建，则直接返回这个对象；否则尝试创建对象；</li>
<li>创建对象的代码被<code>synchronized</code>同步块包围，目的是防止某个线程在创建对象的过程中，另一个线程也尝试创建对象，而同步对象使用的是<code>Singleton</code>类的<code>class</code>对象；</li>
<li>在<code>synchronized</code>代码块中，我们再次判断对象是否为空，若不为空才创建，否则不创建。为什么需要再次判断呢？比如一个线程<code>A</code>运行到了上方代码的<code>2</code>位置，此时被<code>CPU</code>中断，<code>CPU</code>转而执行另一条线程<code>B</code>，<code>B</code>线程运行到<code>1</code>后，发现对象没有创建，于是开始执行下面的代码，而且没有被<code>CPU</code>暂停，于是它成功创建对象。之后<code>CPU</code>重新恢复<code>A</code>线程的运行，<code>A</code>从中断的地方恢复，也就是代码<code>2</code>处，它继续向下运行，如果没有代码<code>3</code>的判断，<code>A</code>线程将再次创建对象，而为了防止这种情况的发生，才需要再次判断；</li>
</ol>
<p>&emsp;&emsp;使用上面的代码后，我们可以发现，只有在对象没有被创建时，才会运行<code>synchronized</code>代码块，当对象成功创建后，将不会再运行到被同步的代码，这就非常巧妙地解决了懒汉式中不必要的同步问题。但是，这个代码虽然看起来解决了问题，但是它并没有我们看起来那么简单，它存在一个非常隐秘的问题。看下面这句代码：</p>
<figure class="highlight java"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">singleton = <span class="keyword">new</span> Singleton();</span><br></pre></td></tr></table></figure>

<p>&emsp;&emsp;看起来这只是一句代码，但是实际上它被分为三条语句执行：</p>
<ol>
<li>为对象分配内存空间；</li>
<li>初始化这个对象；</li>
<li>将对象的引用赋值给singleton；</li>
</ol>
<p>&emsp;&emsp;这是这条语句的正常执行方式，但是编译器常常会在为了优化执行效率，在不改变最终结果的情况下，修改语句的执行顺序（注意：这里所说的不改变结果，指的是单线程下），而上面这条语句也是如此。编译器有时候会对上面三个步骤进行重排序，将其变为1-&gt;3-&gt;2，也就是先分配空间，然后赋值，最后初始化。这样做对效率会有一定的提升，在单线程的环境下也不会有影响。但是，在多线程环境下却发生了问题。当语句被重排序后，假设线程<code>A</code>创建对象，先执行指令<code>1</code>分配空间，然后由于重排序，执行指令<code>3</code>对变量赋值，此时<code>CPU</code>切换线程，线程<code>B</code>开始执行，而<code>A</code>暂时停止。<code>B</code>线程尝试获取单例对象，执行第一条<code>if</code>语句，发现<code>singleton != null</code>，因为之前线程<code>A</code>执行了语句<code>3</code>后，<code>singleton</code>已经指向了对象，于是线程<code>B</code>直接将对象返回，开始使用。这时候就出现问题了，由于<code>A</code>线程没有执行语句<code>2</code>，对象没有被初始化，所以线程<code>B</code>在使用时读取到的内容将是不确定的。为了解决这个问题，<strong>需要为singleton的声明加上volatile关键字，被这个关键字修饰的变量，创建对象时，JVM将不会对指令重排序</strong>。</p>
<br>

<h3 id="emsp-2-5-静态内部类实现"><a href="#emsp-2-5-静态内部类实现" class="headerlink" title="&emsp; 2.5 静态内部类实现"></a>&emsp; 2.5 静态内部类实现</h3><p>&emsp;&emsp;下面再来介绍另外一种对懒汉式的优化实现，通过内部类来实现延迟初始化：</p>
<figure class="highlight java"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">Singleton</span> </span>&#123;</span><br><span class="line"></span><br><span class="line">    <span class="comment">// 内部类实现单例类</span></span><br><span class="line">    <span class="keyword">private</span> <span class="keyword">static</span> <span class="class"><span class="keyword">class</span> <span class="title">Instance</span> </span>&#123;</span><br><span class="line">        <span class="keyword">private</span> <span class="keyword">static</span> <span class="keyword">final</span> Singleton singletion = <span class="keyword">new</span> Singleton();</span><br><span class="line">    &#125;</span><br><span class="line"></span><br><span class="line">    <span class="comment">// 获取单例对象</span></span><br><span class="line">    <span class="function"><span class="keyword">public</span> <span class="keyword">static</span> Singleton <span class="title">getSingleton</span><span class="params">()</span> </span>&#123;</span><br><span class="line">        <span class="keyword">return</span> Instance.singletion;</span><br><span class="line">    &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>

<p>&emsp;&emsp;上面的代码中，我们在<code>Singleton</code>类中创建了一个内部类，这个内部类被修饰为<code>private</code>，也就是只能在<code>Singleton</code>内部访问。而需要创建的单例对象，被放在了内部类中创建，但是提供了一个全局访问接口<code>getSingleton</code>方法，访问内部类中的单例对象。这样的实现有什么好处呢？这是一种天然的，或者说由<code>JVM</code>实现的懒加载模式。类一般是在第一次被使用时才加载，也就是说，只要我们不使用这个内部类，它就不会被加载，自然也就不会创建这个单例对象。只要当第一次调用<code>getSingleton</code>方法时，方法中执行<code>return Instance.singletion;</code>语句时，内部类才被加载，单例对象才被创建。而且最精妙的是，我们前面已经说过，类加载的过程本身就是线程安全的，由<code>JVM</code>帮我们实现了线程同步，所以也不会出现多个线程同时创建单例对象的问题。</p>
<br>

<h3 id="emsp-2-6-枚举类实现"><a href="#emsp-2-6-枚举类实现" class="headerlink" title="&emsp; 2.6 枚举类实现"></a>&emsp; 2.6 枚举类实现</h3><p>&emsp;&emsp;下面再说说最后一种实现方式——枚举类实现。先看看代码：</p>
<figure class="highlight java"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br></pre></td><td class="code"><pre><span class="line"><span class="comment">// enum枚举类</span></span><br><span class="line"><span class="keyword">public</span> <span class="keyword">enum</span> Singleton &#123;</span><br><span class="line"></span><br><span class="line">    <span class="comment">// 单例对象</span></span><br><span class="line">    SINGLETON;</span><br><span class="line"></span><br><span class="line">    <span class="comment">// 单例对象的熟悉</span></span><br><span class="line">    <span class="keyword">private</span> String name;</span><br><span class="line"></span><br><span class="line">    <span class="comment">// 单例对象的方法</span></span><br><span class="line">    <span class="function"><span class="keyword">public</span> String <span class="title">getName</span><span class="params">()</span> </span>&#123;</span><br><span class="line">        <span class="keyword">return</span> name;</span><br><span class="line">    &#125;</span><br><span class="line">    <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setName</span><span class="params">(String name)</span> </span>&#123;</span><br><span class="line">        <span class="keyword">this</span>.name = name;</span><br><span class="line">    &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>

<p>&emsp;&emsp;学习过枚举类型的应该都知道，枚举类的对象天然就是单例的，而且枚举类除了创建对象的方式不同，也同样能够有属性和方法，也就是说，我们完全可以通过枚举类实现天然的单例。这个单例对象将会在枚举类被加载的时候创建，也就是说这是一种饿汉式。使用枚举类创建单例的效率要高于普通饿汉式，因为枚举的单例是由<code>JVM</code>底层实现的，做了优化。而使用枚举类的第二大好处就是：可以避免反射以及序列化对单例模式的破坏。下面就来说一说。</p>
<br>

<h3 id="emsp-2-7-单例模式的破坏"><a href="#emsp-2-7-单例模式的破坏" class="headerlink" title="&emsp; 2.7 单例模式的破坏"></a>&emsp; 2.7 单例模式的破坏</h3><p>&emsp;&emsp;创建对象除了使用<code>new</code>关键字外，还有另外两种方式：</p>
<ul>
<li>通过反射创建对象；</li>
<li>通过序列化和反序列化创建对象；</li>
</ul>
<p>&emsp;&emsp;先看第一种方式，虽然我们将单例类的构造器设为了<code>private</code>，但是对于反射来说，这并不是问题，因为反射连私有成员也能访问，所以对于单例类，我们还能通过如下代码创建对象：</p>
<figure class="highlight java"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">Singleton</span> </span>&#123;</span><br><span class="line"></span><br><span class="line">    <span class="function"><span class="keyword">private</span> <span class="title">Singleton</span><span class="params">()</span> </span>&#123;</span><br><span class="line">    &#125;</span><br><span class="line"></span><br><span class="line">    <span class="function"><span class="keyword">public</span> <span class="keyword">static</span> <span class="keyword">void</span> <span class="title">main</span><span class="params">(String[] args)</span> <span class="keyword">throws</span> Exception </span>&#123;</span><br><span class="line">        <span class="comment">// 反射创建对象</span></span><br><span class="line">        Singleton singleton = Singleton<span class="class">.<span class="keyword">class</span>.<span class="title">newInstance</span>()</span>;</span><br><span class="line">        System.out.println(singleton);</span><br><span class="line">    &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>

<p>&emsp;&emsp;这种方式，也就破化了单例模式只能创建一个对象的原则。前四种实现单例模式的方式都能通过反射创建对象，但是枚举类型却能够避免这个问题。在<code>Class</code>对象的<code>newInstance</code>方法中，若检测到需要创建的对象的类型是<code>enum</code>类型，就会抛出异常（创建<code>Class</code>对象也会）。</p>
<p>&emsp;&emsp;第二种破坏的方式就是使用序列化，简单来说就是先将对象通过流存入文件中，再通过流从文件中读出，则读出的对象和原来的对象将是两个对象，也就是将原来的对象拷贝了一份。前四种实现单例模式的方式都无法避免这个问题，但是，枚举又可以。枚举类型由<code>JVM</code>底层实现了单例，就算是使用序列化，读出来的对象还是原来那个，而不是拷贝一份。所以，枚举类型实现单例是一种非常安全，又简单的方式。</p>
<br>

<h2 id="三、总结"><a href="#三、总结" class="headerlink" title="三、总结"></a>三、总结</h2><p>&emsp;&emsp;当我们要实现饿汉式，也就是类加载时立即创建对象，此时最好是使用枚举类实现，简单高效；当要实现懒汉式时，则推荐使用静态内部类实现，实现简单，无线程安全问题，而且效率也较高。</p>
<br>

<h2 id="四、参考"><a href="#四、参考" class="headerlink" title="四、参考"></a>四、参考</h2><ul>
<li>《Java并发编程的艺术》</li>
<li><a href="https://www.cnblogs.com/chiclee/p/9097772.html" target="_blank" rel="noopener">https://www.cnblogs.com/chiclee/p/9097772.html</a></li>
</ul>

        </div>

        <blockquote class="post-copyright">
    
    <div class="content">
        
<span class="post-time">
    最后更新时间：<time datetime="2020-03-22T17:00:59.886Z" itemprop="dateUpdated">2020-03-23 01:00:59</time>
</span><br>


        
        世界是个球，前方总有路！
        
    </div>
    
    <footer>
        <a href="http://tewuyiang.gitee.io/blog">
            <img src="/blog/img/avatar.jpg" alt="特务依昂">
            特务依昂
        </a>
    </footer>
</blockquote>

        


        <div class="post-footer">
            
	<ul class="article-tag-list" itemprop="keywords"><li class="article-tag-list-item"><a class="article-tag-list-link" href="/blog/tags/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F/" rel="tag">设计模式</a></li></ul>


            
<div class="page-share-wrap">
    

<div class="page-share" id="pageShare">
    <ul class="reset share-icons">
      <li>
        <a class="weibo share-sns" target="_blank" href="http://service.weibo.com/share/share.php?url=http://tewuyiang.gitee.io/blog/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%EF%BC%881%EF%BC%89%E2%80%94%E2%80%94%E5%8D%95%E4%BE%8B%E6%A8%A1%E5%BC%8F%E8%AF%A6%E8%A7%A3/&title=《设计模式（1）——单例模式详解》 — 博客主页&pic=http://tewuyiang.gitee.io/blog/img/avatar.jpg" data-title="微博">
          <i class="icon icon-weibo"></i>
        </a>
      </li>
      <li>
        <a class="weixin share-sns wxFab" href="javascript:;" data-title="微信">
          <i class="icon icon-weixin"></i>
        </a>
      </li>
      <li>
        <a class="qq share-sns" target="_blank" href="http://connect.qq.com/widget/shareqq/index.html?url=http://tewuyiang.gitee.io/blog/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%EF%BC%881%EF%BC%89%E2%80%94%E2%80%94%E5%8D%95%E4%BE%8B%E6%A8%A1%E5%BC%8F%E8%AF%A6%E8%A7%A3/&title=《设计模式（1）——单例模式详解》 — 博客主页&source=一个未来程序员的博客~~~" data-title=" QQ">
          <i class="icon icon-qq"></i>
        </a>
      </li>
      <li>
        <a class="facebook share-sns" target="_blank" href="https://www.facebook.com/sharer/sharer.php?u=http://tewuyiang.gitee.io/blog/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%EF%BC%881%EF%BC%89%E2%80%94%E2%80%94%E5%8D%95%E4%BE%8B%E6%A8%A1%E5%BC%8F%E8%AF%A6%E8%A7%A3/" data-title=" Facebook">
          <i class="icon icon-facebook"></i>
        </a>
      </li>
      <li>
        <a class="twitter share-sns" target="_blank" href="https://twitter.com/intent/tweet?text=《设计模式（1）——单例模式详解》 — 博客主页&url=http://tewuyiang.gitee.io/blog/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%EF%BC%881%EF%BC%89%E2%80%94%E2%80%94%E5%8D%95%E4%BE%8B%E6%A8%A1%E5%BC%8F%E8%AF%A6%E8%A7%A3/&via=http://tewuyiang.gitee.io/blog" data-title=" Twitter">
          <i class="icon icon-twitter"></i>
        </a>
      </li>
      <li>
        <a class="google share-sns" target="_blank" href="https://plus.google.com/share?url=http://tewuyiang.gitee.io/blog/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%EF%BC%881%EF%BC%89%E2%80%94%E2%80%94%E5%8D%95%E4%BE%8B%E6%A8%A1%E5%BC%8F%E8%AF%A6%E8%A7%A3/" data-title=" Google+">
          <i class="icon icon-google-plus"></i>
        </a>
      </li>
    </ul>
 </div>



    <a href="javascript:;" id="shareFab" class="page-share-fab waves-effect waves-circle">
        <i class="icon icon-share-alt icon-lg"></i>
    </a>
</div>



        </div>
    </div>

    
<nav class="post-nav flex-row flex-justify-between">
  
    <div class="waves-block waves-effect prev">
      <a href="/blog/%E8%AF%B4%E8%AF%B4%E7%BA%A2%E9%BB%91%E6%A0%91%E2%80%94%E2%80%94%E4%B8%8D%E8%B0%88%E6%93%8D%E4%BD%9C%EF%BC%8C%E5%8F%AA%E8%AE%B2%E7%90%86%E8%A7%A3/" id="post-prev" class="post-nav-link">
        <div class="tips"><i class="icon icon-angle-left icon-lg icon-pr"></i> Prev</div>
        <h4 class="title">说说红黑树——不谈操作，只讲理解</h4>
      </a>
    </div>
  

  
    <div class="waves-block waves-effect next">
      <a href="/blog/%E3%80%90%E8%BD%AC%E8%BD%BD%E3%80%91%E4%BA%8C%E5%8F%89%E6%8E%92%E5%BA%8F%E6%A0%91%E5%92%8C%E5%B9%B3%E8%A1%A1%E4%BA%8C%E5%8F%89%E6%A0%91%E8%AF%A6%E8%A7%A3/" id="post-next" class="post-nav-link">
        <div class="tips">Next <i class="icon icon-angle-right icon-lg icon-pl"></i></div>
        <h4 class="title">【转载】二叉排序树和平衡二叉树详解</h4>
      </a>
    </div>
  
</nav>



    




















</article>



</div>

        <footer class="footer">
    <div class="top">
        
<p>
    <span id="busuanzi_container_site_uv" style='display:none'>
        站点总访客数：<span id="busuanzi_value_site_uv"></span>
    </span>
    <span id="busuanzi_container_site_pv" style='display:none'>
        站点总访问量：<span id="busuanzi_value_site_pv"></span>
    </span>
</p>


        <p>
            
                <span><a href="/blog/atom.xml" target="_blank" class="rss" title="rss"><i class="icon icon-lg icon-rss"></i></a></span>
            
            <span>博客内容遵循 <a rel="license noopener" href="https://creativecommons.org/licenses/by-nc-sa/4.0/deed.zh" target="_blank">知识共享 署名 - 非商业性 - 相同方式共享 4.0 国际协议</a></span>
        </p>
    </div>
    <div class="bottom">
        <p><span>特务依昂 &copy; 2015 - 2020</span>
            <span>
                
                Power by <a href="http://hexo.io/" target="_blank">Hexo</a> Theme <a href="https://github.com/yscoder/hexo-theme-indigo" target="_blank">indigo</a>
            </span>
        </p>
    </div>
</footer>

    </main>
    <div class="mask" id="mask"></div>
<a href="javascript:;" id="gotop" class="waves-effect waves-circle waves-light"><span class="icon icon-lg icon-chevron-up"></span></a>



<div class="global-share" id="globalShare">
    <ul class="reset share-icons">
      <li>
        <a class="weibo share-sns" target="_blank" href="http://service.weibo.com/share/share.php?url=http://tewuyiang.gitee.io/blog/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%EF%BC%881%EF%BC%89%E2%80%94%E2%80%94%E5%8D%95%E4%BE%8B%E6%A8%A1%E5%BC%8F%E8%AF%A6%E8%A7%A3/&title=《设计模式（1）——单例模式详解》 — 博客主页&pic=http://tewuyiang.gitee.io/blog/img/avatar.jpg" data-title="微博">
          <i class="icon icon-weibo"></i>
        </a>
      </li>
      <li>
        <a class="weixin share-sns wxFab" href="javascript:;" data-title="微信">
          <i class="icon icon-weixin"></i>
        </a>
      </li>
      <li>
        <a class="qq share-sns" target="_blank" href="http://connect.qq.com/widget/shareqq/index.html?url=http://tewuyiang.gitee.io/blog/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%EF%BC%881%EF%BC%89%E2%80%94%E2%80%94%E5%8D%95%E4%BE%8B%E6%A8%A1%E5%BC%8F%E8%AF%A6%E8%A7%A3/&title=《设计模式（1）——单例模式详解》 — 博客主页&source=一个未来程序员的博客~~~" data-title=" QQ">
          <i class="icon icon-qq"></i>
        </a>
      </li>
      <li>
        <a class="facebook share-sns" target="_blank" href="https://www.facebook.com/sharer/sharer.php?u=http://tewuyiang.gitee.io/blog/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%EF%BC%881%EF%BC%89%E2%80%94%E2%80%94%E5%8D%95%E4%BE%8B%E6%A8%A1%E5%BC%8F%E8%AF%A6%E8%A7%A3/" data-title=" Facebook">
          <i class="icon icon-facebook"></i>
        </a>
      </li>
      <li>
        <a class="twitter share-sns" target="_blank" href="https://twitter.com/intent/tweet?text=《设计模式（1）——单例模式详解》 — 博客主页&url=http://tewuyiang.gitee.io/blog/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%EF%BC%881%EF%BC%89%E2%80%94%E2%80%94%E5%8D%95%E4%BE%8B%E6%A8%A1%E5%BC%8F%E8%AF%A6%E8%A7%A3/&via=http://tewuyiang.gitee.io/blog" data-title=" Twitter">
          <i class="icon icon-twitter"></i>
        </a>
      </li>
      <li>
        <a class="google share-sns" target="_blank" href="https://plus.google.com/share?url=http://tewuyiang.gitee.io/blog/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F%EF%BC%881%EF%BC%89%E2%80%94%E2%80%94%E5%8D%95%E4%BE%8B%E6%A8%A1%E5%BC%8F%E8%AF%A6%E8%A7%A3/" data-title=" Google+">
          <i class="icon icon-google-plus"></i>
        </a>
      </li>
    </ul>
 </div>


<div class="page-modal wx-share" id="wxShare">
    <a class="close" href="javascript:;"><i class="icon icon-close"></i></a>
    <p>扫一扫，分享到微信</p>
    <img src="" alt="微信分享二维码">
</div>




    <script src="//cdn.bootcss.com/node-waves/0.7.4/waves.min.js"></script>
<script>
var BLOG = { ROOT: '/blog/', SHARE: true, REWARD: false };


</script>

<script src="//unpkg.com/hexo-theme-material-indigo@latest/js/main.min.js"></script>


<div class="search-panel" id="search-panel">
    <ul class="search-result" id="search-result"></ul>
</div>
<template id="search-tpl">
<li class="item">
    <a href="{path}" class="waves-block waves-effect">
        <div class="title ellipsis" title="{title}">{title}</div>
        <div class="flex-row flex-middle">
            <div class="tags ellipsis">
                {tags}
            </div>
            <time class="flex-col time">{date}</time>
        </div>
    </a>
</li>
</template>

<script src="//unpkg.com/hexo-theme-material-indigo@latest/js/search.min.js" async></script>






<script async src="//busuanzi.ibruce.info/busuanzi/2.3/busuanzi.pure.mini.js"></script>



<script>
(function() {
    var OriginTitile = document.title, titleTime;
    document.addEventListener('visibilitychange', function() {
        if (document.hidden) {
            document.title = '人呢，怎么不见了！';
            clearTimeout(titleTime);
        } else {
            document.title = '(つェ⊂)咦!欢迎回来!';
            titleTime = setTimeout(function() {
                document.title = OriginTitile;
            },2000);
        }
    });
})();
</script>



</body>
</html>
